### РЯЗАНСКИЙ ГОСУДАРСТВЕННЫЙ РАДИОТЕХНИЧЕСКИЙ УНИВЕРСИТЕТ

# РЕАЛИЗАЦИЯ ЦИФРОВЫХ ПРОТОКОЛОВ ПЕРЕДАЧИ ИНФОРМАЦИИ И СИСТЕМ НА КРИСТАЛЛЕ НА ПЛИС

Методические указания к лабораторным работам



Рязань 2016

#### УДК 681.3.06

Реализация цифровых протоколов передачи информации и систем на кристалле на ПЛИС: методические указания к лабораторным работам / Рязан. гос. радиотехн. ун-т; сост. И.С. Холопов. Рязань, 2016. 48 с.

Содержат описание синтеза на ПЛИС систем передачи данных по интерфейсам SPI, I2C, VGA, DVI и LVDS, а также основ синтеза процессоров семейства Nios II на ПЛИС фирмы «Altera» в средах Quartus II и Eclipse.

Работа 1 предназначена для бакалавров направления 11.03.01 — «Радиотехника», изучающих дисциплину «Проектирование цифровых устройств на ПЛИС»; работы 1-4 — для магистрантов направления 11.03.01 — «Радиотехника», изучающих дисциплину «Основы проектирования систем на ПЛИС».

Автор выражает благодарность студентам гр. 210 Бирюкову Антону и Макаркину Павлу за помощь в разработке макетов к лабораторным работам по изучению протоколов SPI и I2C.

Табл. 10. Ил. 24. Библиогр.: 16 назв.

Цифровой конечный автомат, протоколы SPI, I2C, VGA, DVI, LVDS, система на кристалле, процессор Nios II

Печатается по решению редакционно-издательского совета Рязанского государственного радиотехнического университета.

Рецензент: кафедра РТС РГРТУ (зав. кафедрой В.И. Кошелев)

Реализация цифровых протоколов передачи информации и систем на кристалле на ПЛИС

Составитель X о л о п о в Иван Сергеевич

Редактор Р.К. Мангутова Корректор С.В. Макушина

Подписано в печать 12.04.16. Формат бумаги 60 x 84 1/16. Бумага писчая. Печать трафаретная. Усл. печ. л. 3,0. Тираж 50 экз. Заказ

Рязанский государственный радиотехнический университет. 390005, Рязань, ул. Гагарина, 59/1. Редакционно-издательский центр РГРТУ.

### Лабораторная работа № 1 Изучение протокола передачи данных SPI

**Цель работы:** синтез цифрового автомата для реализации информационного обмена с микросхемами, поддерживающими интерфейс SPI (читается как «эс-пи-ай»), на примере цифрового трехосного акселерометра ADXL345 фирмы Analog Devices.

#### ТЕОРЕТИЧЕСКАЯ ЧАСТЬ

Интерфейс **SPI** (от англ. Serial Peripherial Interface – последовательная подключения внешних устройств) шина для разработан компанией Motorola. SPI – дуплексный синхронный протокол последовательного обмена данными, реализованный по принципу Master - Slave (ведущий - ведомый). Обязательным условием перелачи ланных по шине SPI является генерация синхронизации SCLK (Serial Clock). Этот сигнал имеет право генерировать только Master. Данные по протоколу SPI передаются последовательными байтами, бит за битом, начиная со старшего бита. Максимальная длина шины SPI может достигать 5 м, а максимальная частота передаваемых по ней тактовых импульсов - 10 МГц. быстродействующих микросхемы аналого-цифровых преобразователей (АЦП) поддерживают частоту тактовых импульсов шины SPI до 100 МГц.

Существует два основных типа подключения к шине SPI:

- 1) подключение одного устройства Slave;
- 2) подключение нескольких устройств Slave.

Электрическая схема подключения первого типа приведена на рис. 1. На время передачи или приема данных необходимо, чтобы в линии  $\overline{SS}$  (от англ. Slave Select — выбор Slave) действовал низкий логический уровень. В противном случае устройство неактивно. Master



передает данные по линии MOSI (от англ.  $Master\ Output -> Slave\ Input$  — выход  $Master\ ->$  вход устройства) синхронно с сигналом SCLK (от англ.  $Serial\ clock$  — последовательный тактовый сигнал), а Slave принимает биты данных по фронту принятого сигнала синхронизации (переднему или заднему в зависимости от режима работы). Устройство отправляет свои данные по линии MISO (от англ.  $Master\ Input-> Slave\ Output$  — вход  $Master\ ->$  выход устройства).

Электрическая схема подключения второго типа (параллельное подключение n устройств) приведена на рис. 2.



Все сигнальные линии, кроме  $\overline{SS}$ , соединяются параллельно, а *Master* переводом в состояние логического нуля сигнала  $\overline{SS}_i$ ,  $i=\overline{1,n}$ , определяет, с каким устройством он будет обмениваться данными. Главным недостатком такого подключения является необходимость в дополнительных линиях для адресации, так как общее число линий связи для подключения n устройств равно (3+n).

Альтернативные обозначения линий связи MOSI, MISO, SCLK и  $\overline{\rm SS}$ , используемые разными производителями микросхем, сведены в табл 1

Таблица 1. Варианты обозначения сигнальных линий шины SPI

| - mora      |                            |               |  |  |  |  |  |  |  |
|-------------|----------------------------|---------------|--|--|--|--|--|--|--|
| Основное    | Альтернативное обозначение |               |  |  |  |  |  |  |  |
| обозначение | Master                     | Slave         |  |  |  |  |  |  |  |
| MOSI        | DO, SDO, DOUT              | DI, SDI, DIN  |  |  |  |  |  |  |  |
| MISO        | DI, SDI, DIN               | DO, SDO, DOUT |  |  |  |  |  |  |  |
| SCLK        | DCLOCK, SPC, CLK, SCK      |               |  |  |  |  |  |  |  |
| SS          | CS                         |               |  |  |  |  |  |  |  |

Протокол передачи SPI заключается в выполнении операции побитного ввода и вывода данных по фронтам сигнала синхронизации. Установка данных при передаче и чтение при приеме выполняются по противоположным фронтам тактового сигнала SCLK. Существует четыре варианта (режима) работы интерфейса SPI, которые описываются двумя параметрами [1, 2]:



1) CPOL (от англ. *Clock Polarity*) исходный уровень сигнала синхронизации.

Если CPOL = 0, то линия SCLK до начала цикла обмена данными и

после его окончания имеет низкий логический уровень, т.е. первый фронт — нарастающий; если CPOL = 1 — то в линии в состоянии отсутствия обмена данными действует высокий логический уровень, т.е. первый фронт — спадающий (см. рис. 3).

2) СРНА (от англ. *Clock Phase*) — фаза синхронизации. От этого параметра зависит, в какой последовательности выполняется установка и чтение данных: если СРНА = 0, то по первому фронту тактового сигнала SCLK выполняется чтение данных, а по второму — их установка; если СРНА = 1 — то наоборот (см. диаграммы в табл. 2).

Номер режима SPI (от 0 до 3) принято обозначать дибитом (парой бит) CPOL, CPHA (табл. 2).

Таблица 2. Режимы работы SPI

| 1            | аолица 2 | . Режимь | л раооты SPI                         |
|--------------|----------|----------|--------------------------------------|
| Режим<br>SPI | CPOL     | СРНА     | Диаграммы сигналов в линиях шины SPI |
| 0            | 0        | 0        | Установка  SCLK  MOSI  (MISO)        |
| 1            | 0        | 1        | SCLK MOSI (MISO)                     |
| 2            | 1        | 0        | SCLK<br>MOSI<br>(MISO)               |
| 3            | 1        | 1        | SCLK<br>MOSI<br>(MISO)               |

*Master* и *Slave*, работающие в различных режимах, являются несовместимыми, что является недостатком протокола SPI.

При обмене данными по шине SPI возможны прием и чтение как одного байта, так и нескольких байт подряд. Микросхема Slave содержит несколько регистров, в которые можно записывать данные и/или считывать их. Каждый регистр имеет свой адрес A (упрощенно структура регистров Slave приведена на рис. 4); объем регистра, как

правило, составляет 8 бит.



Рис. 4

Рассмотрим обмен данными по SPI в режиме 3 (рис. 5-8), поскольку именно такой режим поддерживается исследуемым в лабораторной работе датчиком ADXL345. На рис. 5 и далее будут использоваться альтернативные обозначения линий SPI относительно микросхемы Slave.

На время передачи данных в соответствии с протоколом SPI необходимо перевести линию CS в низкий логический

уровень. Минимальный интервал времени  $t_{DELAY}$  от заднего фронта CS до первого фронта на линии SCLK приводится в техническом описании (Datasheet) конкретной микросхемы Slave и, как правило, составляет 5...10 нс. Далее на линии Slave SDI(MOSI) Master должен сначала выставить служебный байт, после чего записать в Slave либо считать из него один или несколько байт данных. Служебный байт для ADXL345 состоит из бита чтения или записи R/W (Read/Write: '1' — чтение, '0' — запись), бита множественной или одиночной передачи данных M/S (Multiple/Single: '1' — передача нескольких байт подряд, '0' — передача одного байта) и шести бит адреса регистра Slave, с которым будет происходить обмен данными. По окончании передачи в линии CS устанавливается высокий логический уровень. Стоит отметить, что если в линии CS постоянно действует логический '0' (линия CS соединена с землей), то передача данных невозможна.

Временные диаграммы сигналов в линиях шины SPI при записи одного байта данных  $D = D_7 D_6 D_5 D_4 D_3 D_2 D_1 D_0$  в регистр Slave с адресом  $A = A_5 A_4 A_3 A_2 A_1 A_0$  приведены на рис. 5. При чтении одного байта данных Slave после приема служебного байта по линии SDI выставляет на линии SDO байт данных D, хранящийся в регистре с адресом A. Уровень сигнала в линии SDI в этот момент может быть как высоким, так и низким (рис. 6).

Если необходимо записать данные в несколько соседних регистров подряд (с адресами A, A+1, A+2, ...), то нужно в служебном байте установить бит M/S = '1' и указать адрес первого регистра  $A = A_5 A_4 A_3 A_2 A_1 A_0$ . В этом случае после записи каждого байта данных адрес регистра будет автоматически инкрементироваться (увеличиваться на 1). Временные диаграммы, соответствующие записи двух байт в соседние регистры с адресами A и A+1, приведены на рис. 7. Аналогичным образом может быть организовано чтение нескольких байт данных из соседних регистров (рис. 8).



Рис. 8

Некоторые микросхемы поддерживают трехпроводной протокол SPI, в котором вместо двух разнонаправленных линий SDI и SDO используется одна двунаправленная линия SDIO. Временные диаграммы при записи данных B Slave этом случае четырехпроводному SPI, а при чтении данные D после приема служебного байта А выставляются на той же линии SDIO. Для реализации двунаправленной линии средствами VHDL порт SDIO должен быть объявлен как **inout** (Z-состояние по умолчанию), а линия SDIO подключена к шине питания через подтягивающий резистор. Передача данных при этом возможна только выдачей низкого логического уровня в линию SDIO (замыканием ее на землю), а высокий логический уровень формируется благодаря подтягивающему резистору.

Реализация информационного обмена с любой микросхемой подразумевает внимательное изучение ее технического описания, которое часто приводится только на английском языке. Основные сведения о микросхеме, как правило, приводятся в аннотации (Features) технического описания [3, Р. 1]: для исследуемого трехосного датчика ускорения (акселерометра) ADXL345 приводятся диапазоны измерения ускорения (Full resolution range), напряжение питания (Supply voltage), цена младшего разряда (LSB scale factor, где аббревиатура LSB означает младший значащий бит, Least Significant Bit), поддерживаемые протоколы передачи данных (Interfaces). Далее техническом описании В электрическую схему подключения для выбранного протокола передачи данных [3, Р. 15].

Карта всех регистров и описание назначения каждого из них приводятся в разделах *Register Map* [3, P. 23-27] и *Applications Information* [3, P. 28-36] соответственно.

Регистры управления микросхем часто имеют обозначение CONTROL, CTRL или CTL. Для датчика ADXL345 таким регистром является регистр POWER CTL [3, P. 25] с адресом (Address) 2D в шестнадцатеричной счисления (Hex)системе программирования высокого уровня принято обозначение с префиксом (0x) - 0x2D) или 45 – в десятичной (*Dec*). Тип регистра (*Type*) R/W означает, что для него разрешены как чтение, так и запись данных; значение по умолчанию (после подачи на устройства питания)  $\langle\langle 00000000\rangle\rangle$  = 0x00 приводится в колонке Reset Value. Описание назначения каждого бита данного регистра приводится в подразделе Register 0x2D - POWER CTL [3, P. 25-26]. Бит Link позволяет произвести разделение функций датчика в активном и неактивном режимах: при значении '1' такое разделение возможно, при '0' - нет (в работе выберем последний вариант). лабораторной Высокий логический уровень бита  $AUTO\_SLEEP$  разрешает устройству автоматически переходить в режим пониженного энергопотребления; также установим этом бит в '0'. Бит *Measure* определяет режим работы устройства: '1' — режим измерения, '0' — режим ожидания. Значение '1' в бите *Sleep* устанавливает пониженную частоту дискретизации (темп измерений): значение 1, 2, 4 или 8  $\Gamma$ ц определяется двумя битами *Wakeup* [3, P. 26]. Используемый в лабораторной работе управляющий байт для регистра POWER CTL — 0x08.

Регистр BW\_RATE содержит бит *LOW\_POWER* ('1' — режим низкого энергопотребления с большим уровнем шумов, '0' — нормальный режим) и 4 бита *Rate* [3, P. 25], которые определяют частоту дискретизации датчика.

Установка '1' в биты регистра INT\_ENABLE разрешает различные виды прерываний (см. регистр INT\_SOURCE), а '0' – запрещает.

Регистр DATA FORMAT определяет формат представления данных с датчика в регистрах с адресами 0х32-0х37. Высокий логический уровень бита SELF TEST (самотестирование) изменяет реальные данные с датчика: такой режим иногда может потребоваться при отладке. Бит SPI устанавливает 4-проводной ('0') или 3-проводной ('1') интерфейс SPI. Установка '1' в бит *INT INVERT* инвертирует уровни прерываний в регистре прерываний INT SOURCE. Бит FULL RES определяет изменение разрешения акселерометра: при '0' разрядность акселерометра фиксирована – независимо от диапазона измерений отсчет выходного сигнала по каждой оси представляется 10-ю битами (т.е. при расширении диапазона измерений увеличивается цена младшего разряда LSB); при '1' число уровней квантования увеличивается с ростом динамического диапазона (цена LSB не меняется). Бит Justify задает выравнивание дополнительного кода отсчета ускорения (10-13 бит в зависимости от диапазона измерений) по правому ('0') или левому ('1') краю в паре байт выходного сигнала для каждой оси. В первом случае знак ускорения дублируется на свободные старшие разряды, во втором случае младшие разряды дополняются не значащими нулями (см. пример для 10-разрядного кода "10 10001000" в табл. 3). Биты Range задают диапазон измерений акселерометра.

Таблица 3. Влияние бита *Justify* на формат данных акселерометра

| Бит     | 16 бит в паре выходных регистров Datax0 (младший байт) и Datax1 (старший байт), $x = \{X, Y, Z\}$ |   |   |   |   |   |        |   | И |   |   |   |   |   |   |   |
|---------|---------------------------------------------------------------------------------------------------|---|---|---|---|---|--------|---|---|---|---|---|---|---|---|---|
| Justify | Datax1                                                                                            |   |   |   |   |   | Datax0 |   |   |   |   |   |   |   |   |   |
| 0       | 1 1 1 1 1 1 1 <b>1 0</b>                                                                          |   |   |   |   | 1 | 0      | 0 | 0 | 1 | 0 | 0 | 0 |   |   |   |
| 1       | 1                                                                                                 | 0 | 1 | 0 | 0 | 0 | 1      | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |

Перед чтением данных из регистров с адресами 0x32-0x37 необходимо проверить, записаны ли в них новые данные от встроенного в акселерометр АЦП, прочитав бит готовности данных *DATA READY* регистра прерываний INT\_SOURCE.

С учетом технического описания ADXL345 для чтения данных можно использовать следующий алгоритм работы цифрового конечного автомата (ЦКА) на ПЛИС:

- 1) инициализация регистров управления POWER\_CTL, BW\_RATE, INT\_ENABLE и DATA\_FORMAT, т.е. запись в них управляющих слов для обеспечения требуемого режима работы;
- 2) чтение регистра прерываний INT\_SOURCE: если бит  $DATA\_READY = '1'$ , выполнить п. 3), в противном случае повторять п. 2):
- 3) чтение данных об ускорении вдоль осей X, Y и Z из регистров с адресами 0x32-0x37 в буферный регистр ПЛИС;
  - 4) выдача данных по завершении чтения;
- 5) ожидание следующего периода чтения данных с датчика (величина периода определяется требуемой частотой дискретизации) и переход к п. 2).

Граф состояний ЦКА, реализующего SPI-Master для датчика ADXL345 в соответствии с приведенным выше алгоритмом, показан на рис. 9. ЦКА содержит 5 состояний: sleep — ожидание, init — инициализация,  $int\_status$  — чтение состояния регистра прерываний,  $read\_data$  — чтение данных в буферный регистр ПЛИС,  $d\_ready$  — анализ готовности данных.



Рис. 9

На рис. 9 цифрами на направлениях смены состояний ЦКА обозначены следующие условия:

- 1) инициализация не завершена;
- инициализация завершена, наступил новый цикл чтения данных;
- 3) не выполнены условия 1) и 2);
- 4) инициализирующий байт записан в регистр управления;
- 5) условие 4) не выполнено;
- 6) прочитано содержимое регистра прерываний;

- 7) условие 6) не выполнено;
- 8) данные готовы;
- 9) данные не готовы;
- 10) прочитаны все байты данных;
- 11) условие 10) не выполнено.

#### Домашнее задание

- 1. Изучить основы передачи данных по протоколу SPI.
- 2. Ознакомиться с техническим описанием интегральной микросхемы ADXL345 [3]: схема включения; порядок обмена информацией по протоколу SPI; адреса регистров управления и значения управляющих байт для них; принцип декодирования кода ускорения.
- 3. По Datasheet определить значения управляющих байт при следующих требованиях к акселерометру: байт для регистра  $POWER\_CTL 0x08 = 8$  (нормальное энергопотребление), частота дискретизации  $-100~\Gamma$ ц, диапазон измерений  $-\pm 2g$ , разрешено прерывание по готовности данных, выравнивание данных по левому краю для нечетного варианта и по правому для четного; фиксированная разрядность 10~ бит, количество линий шины SPI-4.
- 4. По графу состояний рис. 9 составить программы на языке описания аппаратуры VHDL для чтения данных с акселерометра ADXL345 в соответствии с требованиями пп. 1-3 порядка выполнения работы.

### ПОРЯДОК ВЫПОЛНЕНИЯ РАБОТЫ

- 1. Моделирование обмена данными по протоколу SPI на примере трехосного акселерометра ADXL345
  - 1.1. Создать новый проект с именем spi master.
  - 1.2. Составить описание пакета, в котором объявить:
  - 1) перечислимый тип spi states с состояниями ЦКА SPI-Master;
  - 2) подтип *byte* как тип **natural** в диапазоне 0-255;
  - 3) константы подтипа *byte*:
- BW\_RATE, POWER\_CTRL, INT\_ENABLE и DATA\_FORMAT для адресов регистров управления и INT\_SOURCE для адреса регистра прерываний;
- WORD\_BW\_RATE, ..., WORD\_DATA\_FORMAT для управляющих слов длиной 1 байт, которые будут записаны в соответствующие регистры при инициализации;
- ALL\_INIT\_REGS и ALL\_READ\_REGS для общего количества инициализируемых регистров и регистров с данными от датчика;
  - DATA0 для адреса первого регистра с данными.

- 4) константу DISCRETE типа **natural**, определяющую период дискретизации, выраженный в периодах частоты тактового сигнала *sclk*.
  - 1.3. Составить описание сущности SPI-*Master* с портами:
  - 1) clock: in std logic; -- тактовый сигнал кварцевого генератора;
  - 2) cs: out std logic; -- линия CS;
  - 3) sclk: out std logic; -- линя SCLK;
  - 4) sdi: **out std logic**; -- линия SDI Slave;
  - 5) sdo: in std logic; -- линия SDO Slave;
- 8) выходной вектор объемом 2 байта для вывода кода ускорения по одной из осей, указанной преподавателем.
  - 1.4. В декларативной части архитектуры объявить:

тип *adxl\_regs* как массив байт длиной ALL\_INIT\_REGS (нумерация элементов массива возрастающая, начиная с нуля);

тип data\_regs как массив из ALL\_READ\_REGS элементов типа std\_logic\_vector(7 downto 0) с аналогичной нумерацией элементов;

константы типа adxl regs:

INIT\_ADDRESES = (BW\_RATE, ..., DATA\_FORMAT) и CTRL\_DATA = (WORD\_BW\_RATE, ..., WORD\_DATA\_FORMAT). Объявить сигналы типа **std logic**:

new\_data - признак готовности новых данных в датчике ('1' - данные готовы);

init\_ready – признак окончания инициализации ('1' – инициализация завершена);

clk – внутренний тактовый сигнал (выход сигнала делителя);

sclk\_ena – признак разрешения выдачи clk в линию sclk ('1' – выдача разрешена);

wd – сигнал от внутреннего таймера о начале нового цикла чтения данных ('1' – начало нового цикла);

cnt – сигнал разрешения счета бит ('1' – счет разрешен).

-- сигналы состояний ЦКА;

signal state, next\_state: spi\_states;

-- счетчик бит;

signal n bit: natural range 0 to 8\*(ALL READ REGS + 1) - 1;

-- число проинициализарованных регистров;

signal num\_init: natural range 0 to ALL\_INIT\_REGS;

-- число прочитанных регистров;

signal num\_read: natural range 0 to ALL\_READ\_REGS;

1.5. Составить описание проекта делителя частоты с настроечной константой. Выходной сигнал делителя должен представлять меандр с высоким логическим уровнем в первом полупериоде. Подключить к проекту компонент делителя.

```
1.6. Для синтеза процессов, реализующих SPI-Master, можно
воспользоваться шаблоном, приведенным ниже.
       -- формирование сигнала в линии SCLK
       sclk \le <\piог. vровень> when sclk ena = '0' else clk:
       -- процесс определения следующего состояния ЦКА
       process (state, wd, init ready, n bit, new data, num read)
       begin
       case (state) is
        when sleep \Rightarrow if (init_ready = '0') then next_state \iff coctoshue\Rightarrow:
                         elsif (wd = '1') then next state <= <состояние>:
                          else next state <= <состояние>:
                       end if:
        when init \Rightarrow if (n bit = 15)
                          then next state <= <cостояние>;
                          else next state <= <cостояние>:
                       end if:
        when int status \Rightarrow if (n bit = 15)
                              then next state <= <cостояние>;
                               else next state <= <состояние>;
                            end if:
        when d ready =>
          if (new data = '0') then next state \leq \leqсостояние\geq;
          else next state <= <состояние>:
          end if:
        when read data => if (n bit = (8*(ALL READ REGS + 1) - 1))
                              then next state <= <cостояние>;
                              else next state <= <состояние>;
                            end if:
        when others => null;
       end case:
       end process;
       -- описание счетчика бит по заднему фронту clk с асинхронным
       -- сбросом по низкому логическому уровню сигнала cnt
       process (<список чувствительности>)
        <операторы для реализации счетчика>;
       -- процесс формирования сигнала разрешения счета бит
       process (<список чувствительности>)
       begin
       if (clk'event and clk = '0') then
        case (state) is
         when sleep \Rightarrow cnt \Leftarrow '0':
```

```
when init => cnt <= '1':
  when int status => cnt <= '1':
  when read data => cnt <= '1';
  when d ready \Rightarrow cnt \Leftarrow '0':
  when others => null:
end case;
end if:
end process;
-- внутренний таймер и смена состояний ЦКА
process (<список чувствительности>)
variable cycle: natural range 0 to DISCRETE;
begin
if (clk'event and clk = '1') then
 state <= next state;
 if cvcle = 0 then wd \le '1':
  else wd \le 0':
 end if:
 if cycle = DISCRETE - 1 then cycle := 0;
  else cycle := cycle + 1;
 end if:
end if:
end process;
-- процесс формирования сигналов в линиях SDI и CS
process (clock, state, n bit, num init, num read)
-- регистры для временного хранения данных
variable buf 8: std logic vector(7 downto 0);
variable buf 16: std logic vector(15 downto 0);
begin
 if (clock'event and clock = '1') then
 case (state) is
                  sdi <= '0'; cs <= '1'; sclk ena <= '0';
 when sleep =>
 when init =>
          cs <= '0'; sclk ena <= '1';
          buf 16 := <биты R/W и M/S> &
          conv std logic vector(INIT ADDRESES(num init), 6)&
          conv std logic vector(CTRL DATA(num init), 8);
          sdi \le buf 16(15 - n bit);
 when int status => cs <= '0'; sclk ena <= '1';
                     buf 8 := <биты R/W и M/S> &
                     conv std logic vector(<адрес>, 6);
                     sdi \le buf 8(7 - n bit);
```

```
when read data \Rightarrow cs \iff '0': sclk ena \iff '1':
                      buf 8 := <биты R/W и M/S> &
                      conv std logic vector(<адрес>, 6);
                      sdi \le buf 8(7 - n bit):
 when d ready \Rightarrow cs \Leftarrow '1'; sclk ena \Leftarrow '0'; sdi \Leftarrow '0';
 when others \Rightarrow null:
 end case:
end if:
end process;
-- процесс, выполняющий чтение данных с линии SDO Slave
-- и формирование управляющих сигналов
process (<список чувствительности>)
variable data: data regs;
begin
if (clk'event and clk = '1') then
 case (state) is
   when sleep \Rightarrow num read \iff 0; new data \iff 0';
                   <выход> <= <старший байт> & <младший байт>;
   when init \Rightarrow if (n bit = 15) then
                    if (num init = ALL INIT REGS - 1)
                      then init ready <= <лог. уровень>;
                    else init ready <= <лог. уровень>;
                        num init \leq num init + 1;
                    end if:
                 end if:
 when int status => if (n bit =<номер бита>)
                          then new data <= sdo;
                      end if:
 when read data => if (n bit ><номер бита>)
                        then data(num read)(7 - (n \text{ bit rem } 8)) := \text{sdo};
                        if (n bit rem 8 = 7)
                         then num read \leq num read + 1;
                        end if:
                      end if:
 when others \Rightarrow null;
 end case:
end if:
end process;
```

1.7. Задать настроечную константу делителя равной 8, константу DISCRETE = 200. Откомпилировать проект; при наличии в тексте программы ошибок внести исправления.

1.8. Выполнить моделирование работы проекта. Для упрощения отладки добавить в \*.vwf-файл помимо входных и выходных сигналов сущности текущее состояние ЦКА, служебные сигналы и переменную cycle. Задать частоту сигнала с кварца 25 МГц, время моделирования — 200 мкс. По умолчанию установить на линии SDO логический '0'. Запустить моделирование. Убедиться в правильности записи заданных байт в регистры управления. Выполнить перезапись \*.vwf-файла результатами моделирования (П.1). Сымитировать во втором по порядку состоянии ЦКА int\_status ответ от датчика о готовности данных. Выполнить моделирование. Сымитировать в состоянии read\_data ненулевой ответ от Slave и снова выполнить моделирование. Проанализировать работу синтезированного ЦКА.

## 2. Проверка работы проекта с датчиком ADXL345

- 2.1. Установить коэффициент деления делителя таким, чтобы обеспечить скорость обмена по шине SPI 1 Мбит/с; вернуть исходное значение константы DISCRETE.
- 2.2. Подключить входные и выходные сигналы проекта к выводам ПЛИС отладочного комплекта. Линиям шины SPI назначить номера выводов ПЛИС в соответствии со схемой подключения датчика ADXL345 к контактам разъема JP 1 (GPIO 0) или JP2 (GPIO 1). Выходной сигнал подключить к красным и зеленым светодиодам LEDR7 LEDR0 и LEDG7 LEDG0. Откомпилировать проект заново.
- 2.3. Не подключая плату с ПЛИС к ПЭВМ, соединить контакты разъема датчика с контактами разъема ЈР 1 (GPIO 0): VCC с контактом 3,3 В, GND с контактом GND. Показать преподавателю результаты подключения выводов датчика. Предельное напряжение обратной полярности для ADXL345 составляет всего -0,3 В, поэтому при неправильной коммутации выводов VCC и GND датчик будет поврежден.
  - 2.4. Включить отладочный макет. Загрузить проект в ПЛИС.
- 2.5. Наблюдать на светодиодах изменение сигнала ускорения. Повернув датчик таким образом, чтобы нормированный модуль проекции вектора ускорения свободного падения на выбранную ось чувствительности составил  $\pm 1$  (можно воспользоваться рисунками Figure 57 и 58 Datasheet [3]), объяснить влияние выбранного бита Justify на форму представления выходных данных.

### 3. Индикация ускорения на семисегментных индикаторах

3.1. Добавить в программу компоненты трехвходового мультиплексора для переключения между данными с каждой из осей чувствительности датчика, семисегментного индикатора (ССИ) и блок декодера ускорения (код ускорения ADXL в соответствии с *Datasheet* 

перевести в доли д и выдать значение в 100 раз большее).

- 3.2. Изменить темп чтения данных с акселерометра на 2 или 5 Гц.
- 3.3. Загрузить скомпилированный проект в ПЛИС. Убедиться в правильности его работы (индикации ускорений со знаком) для всех осей чувствительности.
  - 3.4. По окончании работы выключить отладочный макет.

### Лабораторная работа № 2 Изучение протокола передачи данных I2C

**Цель работы:** синтез цифрового автомата для реализации информационного обмена с микросхемами, поддерживающими интерфейс I2C (читается как «ай-ту-си»), на примере цифрового трехосного акселерометра ADXL345 фирмы Analog Devices.

Интерфейс I2C (от англ. Inter-Integrated Circuit – схема компанией Philips. внутренней связи) разработан 12C последовательный полудуплексный протокол передачи данных. Для обмена используются две линии связи: линия данных SDA (от англ. Serial Data - последовательные данные) и линия тактового сигнала SCL (от англ. Serial Clock - тактовый сигнал). Шина I2C, как и SPI, организована по принципу Master-Slave. Протокол I2C поддерживает информацией на расстоянии до 30...50 см. К одной двухпроводной шине одновременно может быть подключено до 127 устройств, различающихся 7-битным адресом (для микросхем с 10-битным адресом максимальное количество равно 1023).

Подключение к шине I2C (рис. 10) можно рассматривать как соединение выводов микросхем с открытым коллектором (высокоимпедансное Z-состояние) по схеме «монтажное И»: поскольку все устройства Slave включены параллельно, то выдача логического нуля хотя бы одной микросхемой на линии SDA или SCL приводит к появлению низкого логического уровня во всей линии.



Передача/прием данных по шине I2C осуществляется переводом линии SDA в низкий логический уровень. Тактовый сигнал SCL имеет право генерировать только *Master*. Максимальная частота тактовых

импульсов составляет 100 или 400 кГц. При передаче логической '1' запрещено генерировать сигнал сильной '1', поскольку, если одно из устройств *Slave* этот момент установит на линии логический '0', это приведет к короткому замыканию (см. рис. 10) и повреждению *Master* или *Slave*. Поэтому высокий логический уровень в линиях шины I2C следует задавать только их переводом в Z-состояние, при этом сигнал логической '1' формируется с помощью подтягивающих резисторов R.

Начало передачи определяется сигналом *Start* (S) — переходом линии SDA из '1' в '0' при высоком логическом уровне сигнала в линии SCL (рис. 11, а). Окончание передачи определяется сигналом *Stop* (P) — переходом сигнала SDA из '0' в '1' при высоком логическом уровне SCL (рис. 11, б). При передаче информации от *Slave Master* генерирует тактовый сигнал в линии SCL и считывает биты с линии SDA (рис. 11, в). *Slave* выставляет новый бит при SCL = '0', Master считывает данный бит при SCL = '1', т.е. считывание производится не по фронту, а по уровню тактового сигнала. Этим объясняются меньшие помехоустойчивость и скорость передачи данных шины I2C по сравнению с шиной SPI. При передаче информации от *Master* к *Slave Мaster* генерирует тактовый сигнал в линии SCL и выдает биты в линию SDA. *Slave* считывает данные по высокому логическому уровню в линии SCL. Длительности '0' и '1' в периоде тактового сигнала SCL рекомендуется устанавливать в соотношении 70/30 или 60/40.

Таким образом, изменение данных на линии SDA может выполняться только при низком логическом уровне в линии SCL; если SCL = '1', то происходит их чтение. Если изменения на линии SDA происходят при SCL = '1', то это соответствует служебной команде *Start* или *Stop*.

Данные по протоколу I2С передаются последовательными байтами, бит за битом, начиная со старшего бита [4, С. 5-34]. Первый байт всегда направляется от Master к Slave и представляет собой физический адрес устройства SAD (от англ. Slave address) длиной 7 бит + 8-й бит направления передачи R/W (Read/Write): ('0' — запись в устройство, '1' — чтение из устройства). Бит R/W определяет, что будет делать Slave при передаче следующего байта: передавать или принимать данные.

Если байт данных был принят Slave корректно, то устройство вырабатывает бит подтверждения АСК (от англ. Acknowledge подтверждение, ответ). В некоторых Datasheet бит ACK может быть обозначен как SAK (от англ. Slave Acknowledge). Если адрес SAD. переданный Master, совпал с адресом Slave, то вырабатывается ACK = '0' и передача данных продолжается. Если ACK = '1', это означает, что Slave с данным адресом на линии не обнаружен, либо неправильно принял байт со своим адресом, либо неработоспособен. При записи данных в Slave следующие за SAD байты могут быть байтами с адресами регистров Slave (обозначаются как SUB от англ. Sub-address) или байтами данных DATA. После корректного приема каждого байта Slave также отвечает байтом подтверждения АСК = '0'. При чтении данных из Slave сигнал подтверждения о приеме байта (бит '0') должен вырабатывать Master. Этот сигнал обозначают как MAK (Master acknowledge). После приема последнего байта перед выработкой сигнала Stop Master должен сообщить Slave, что больше не будет принимать от него данные. Для этого он вырабатывает на линии SDA сигнал высокого логического уровня NACK = '1' (от англ. *No acknowledge* – нет ответа).

Если необходимо записать в Slave несколько байт в соседние регистры, то выполняется передача адреса SUB первого регистра, а затем подряд n байт данных. Адрес SUB при записи каждого нового байта данных будет автоматически инкрементироваться. Аналогичное справедливо и для чтения нескольких байт из Slave.

Например, для записи двух байт в соседние регистры необходимо:

- 1) сформировать сигнал S;
- 2) отправить 7 бит адреса Slave SAD и бит записи R/W = '0';
- 3) получить бит подтверждения АСК;
- 4) отправить байт адреса регистра *Slave* SUB, в который требуется записать первый байт данных;
  - 5) получить бит подтверждения АСК;
- 6) отправить байт данных DATA для записи в регистр с адресом SUB;
  - 7) получить бит подтверждения АСК;
- 8) отправить второй байт данных для записи в регистр с адресом SUB+1;
  - 9) получить бит подтверждения АСК;
  - 10) после приема последнего АСК сформировать сигнал Р.

Для чтения нескольких двух байт из соседних регистров *Slave* необходимо:

- 1) сформировать сигнал S;
- 2) отправить 7 бит адреса Slave SAD и бит записи R/W = '0';

- 3) получить бит подтверждения АСК;
- 4) отправить байт адреса регистра *Slave* SUB, из которого требуется считать первый байт данных;
  - 5) получить бит подтверждения АСК;
  - 6) сформировать сигнал повторного старта **RS**;
  - 7) отправить 7 бит адреса Slave SAD и бит чтения R/W = '1';
  - 8) получить бит подтверждения АСК;
  - 9) получить байт данных DATA из регистра с адресом SUB;
  - 10) сформировать бит подтверждения МАК;
  - 11) получить байт бит данных DATA из регистра с адресом SUB + 1;
  - 12) после приема последнего байта данных сформировать бит NMAK;
  - 13) сформировать сигнал Р.

Служебный сигнал **RS** (от англ. *Repeated Start* – повторный старт) аналогичен сигналу **S** и разделяет циклы записи и чтения при чтении данных из *Slave*.

В табл. 4. и 5 указан порядок выработки сигналов *Master* и *Slave*, а на рис. 13 и 14 приведены соответственно временные диаграммы, демонстрирующие запись и чтение двух байт подряд, начиная с регистра с адресом SUB.

В табл. 6 приведен пример с порядком выставки бит на шине SDA для записи двух байт (0xF0 и 0x77) в регистры с адресами 0x0A и 0x0B, а в табл. 7 – для чтения двух байт (0x03 и 0x0D) из регистров с адресами 0x0F и 0x10. Адрес *Slave* принят равным 0x63.

Устройства Slave с низкой скоростью записи данных (например, электрически репрограммируемые постоянные запоминающие устройства EEPROM) требуют некоторого времени для записи байта данных, поэтому после ответа АСК Slave может удерживать на линии SCL низкий логический уровень (показан на рис. 12 утолщенной линией), не позволяя Master генерировать новые тактовые импульсы (показаны пунктиром): по принципу монтажного И на линии SCL в этом случае будет действовать логический '0'. Поэтому при работе с такими Slave после записи байта данных Master должен анализировать состояние линии SCL ('0' – занята, '1' – свободна) и при необходимости задерживать передачу следующего байта.





В публицистическом стиле принцип работы протокола I2C подробно описан в интернет-источниках [5] и [6].

Логика работы цифрового автомата для реализации I2C *Master* для чтения данных с датчика ADXL345 схожа с принципом работы ЦКА SPI *Master*, рассмотренного в лабораторной работе № 1: сначала необходимо записать управляющие слова в контрольные регистры, а затем, анализируя бит готовности данных в регистре INT\_SOURCE, с заданным интервалом дискретизации считывать данные. Особенностью ЦКА I2C *Master* является наличие дополнительных состояний: обработки бита АСК и генерации/анализа сигналов S, P, RS, ACK и NACK (рис. 15).



Рис. 15

Поскольку чтение содержимого нескольких соседних регистров быстрее, чем чтение одного байта из каждого регистра, в работе будут считываться 8 байт данных из регистров от INT\_SOURCE до DATAZ1 (т.е. также будут прочитаны данные из регистра DATA\_FORMAT).

#### Домашнее задание

- 1. Изучить основы передачи данных по протоколу I2C.
- 2. Определить по *Datasheet* значения управляющих байт при следующих требованиях к акселерометру: байт для регистра POWER\_CTL -0x08=8 (нормальное энергопотребление), частота дискретизации -100  $\Gamma$ ц, диапазон измерений  $-\pm 2g$ , разрешено прерывание по готовности данных, выравнивание данных по левому краю для нечетного варианта и по правому для четного; фиксированная разрядность 10 бит, разрешен интерфейс 12C.
- 3. По графу состояний рис. 15 и в соответствии с логикой работы цифрового автомата из лабораторной работы № 1 составить программы на языке описания аппаратуры VHDL для чтения и обработки данных с датчика акселерометра ADXL345 в соответствии с требованиями пп. 1 и 2 порядка выполнения работы.

#### ПОРЯДОК ВЫПОЛНЕНИЯ РАБОТЫ

- 1. Моделирование обмена данными по протоколу I2C на примере трехосного акселерометра ADXL345
  - 1.1. Создать новый проект с именем *i2c master*.
- 1.2. Повторить п. 1.2 лабораторной работы № 1 со следующими изменениями.

Вместо типа spi states объявить тип i2c states с состояниями:

sleep - ожидание;

start – генерация сигнала S;

slave addr – передача адреса Slave;

data tx rx – прием/передача байт данных;

stop – генерация сигнала Р;

sak – анализ ответа АСК от Slave;

error – состояние ошибки: отсутствие ответа ACK от Slave;

mak – генерация сигнала MAK Master;

nack - генерация сигнала NACK Master;

rs – генерация сигнала повторного старта.

- В пакете также объявить константу адреса *Slave* SLAVE ADRESS (без младшего бита SAD0) типа *byte*.
  - 1.3. Составить описание сущности I2C-Master с портами:
  - 1) clock: in std\_logic; -- тактовый сигнал кварцевого генератора;
  - 2) cs: **out std\_logic**; -- сигнал выбора режима I2C для ADXL345;
  - 3) scl: inout std\_logic; -- линя SCL;
  - 4) sda: inout std\_logic; -- линия SDA;
  - 5) sad0: **out std\_logic**; -- младший бит адреса *Slave*;
  - 6) err: out std\_logic; -- признак состояния error;
- 7) выходной вектор объемом 2 байта для вывода кода ускорения по одной из осей, указанной преподавателем.
  - 1.4. В декларативной части архитектуры объявить:

тип  $adxl\_regs$  длиной ALL\_INIT\_REGS+1 байт;

тип  $data\_regs$  — длиной ALL\_READ\_REGS+1 байт;

константы типа adxl regs:

INIT\_ADDRESES = (BW\_RATE, ..., INT\_SOURCE) и

 $CTRL_DATA = (WORD_BW_RATE, ..., 0).$ 

Объявить сигналы типа std\_logic:

new\_data – признак готовности новых данных в датчике ('1' – данные готовы);

init – признак инициализации ('1' – инициализация);

clk – внутренний тактовый сигнал (1-й выход делителя);

clk\_for\_scl – сигнал для выдачи в линию SCL (2-й выход делителя); gen scl – признак разрешения выдачи clk\_for\_scl в линию SCL

```
('1' – выдача разрешена);
      free – признак свободной линии SCL ('1' – линия свободна);
      rw – чтение/запись ('1'/'0');
      аск – признак ответа АСК:
      wd - сигнал от внутреннего таймера о начале нового цикла
чтения данных ('1' – начало нового цикла);
      cnt – сигнал разрешения счета бит ('1' – счет разрешен);
      RStart – сигнал для генерации повторного старта RS:
      data adress – признак передачи данных ('1') или адреса ('0').
      Также в архитектуре необходимо объявить:
      сигнал type tx перечислимого типа с элементами tx adr, tx data,
tx stop и tx RS для определения действий Master после приема ответа
ACK (соответственно передача адреса Slave, байта SUB или DATA,
сигнала Р и сигнала RS);
                                                  byte
      сигналы
                num init
                           И
                               num read
                                           типа
                                                         ДЛЯ
                                                               счета
                                                  регистров
проинициализированных
                            И
                                 прочитанных
                                                               Slave
соответственно;
      сигнал для счета бит n bit типа natural range 0 to 7;
      сигналы state и next state для хранения состояний I2C Master.
      1.5. Составить
                      описание
                                  проекта
                                             делителя
настроечной константой и двумя выходами. Сигнал с первого выхода
делителя объявить как out std logic и сформировать на нем меандр с
низким логическим уровнем в первом полупериоде; сигнал со второго
выхода объявить как inout std logic и сформировать на нем меандр с
'Z' состоянием во второй и третьей четвертях периода. Подключить
компонент делителя к проекту.
      1.6. Присвоить требуемые логические уровни выходам sad0 и сs.
      1.7. Для синтеза процессов, реализующих I2C-Master, можно
воспользоваться шаблоном, приведенным ниже.
      -- формирование сигнала в линии SCL
      scl <= clk for scl when gen scl = '1' else 'Z';
      -- процесс смены состояний ЦКА
      process (clk, wd)
      begin
      if clk'event and clk = '0' then
        if wd = '1' then
        state <= start:
```

**else** state <= next state:

end if;
end if;
end process;

```
-- процесс определения следующего состояния ЦКА
process(state, ack, n bit, num read, num init, new data, rw, free, type tx)
begin
case state is
 when sleep => next state <= sleep:
 when start => next state <= <состояние>:
 when slave addr \Rightarrow if (n bit = 7) then next state \iff cостояние\Rightarrow;
                       else next state <= <состояние>:
                       end if:
 when sak \Rightarrow if ack = '0' then
               case (type tx) is
                   when tx adr => next state <= data tx rx;
                   when tx data => next state <= data tx rx;
                   when tx stop => next state <= stop;
                   when tx RS => next state <= RS:
                   when others => null;
               end case:
              else next state <= error;
              end if:
 when mak => next state <= <cостояние>:
 when RS => next state <= <состояние>:
 when nack => next state <= <cостояние>:
 when stop => if free = '0' then next state <= stop;
               else next state <= sleep;
               end if:
 when error => next state <= error;
 when data tx rx \Rightarrow if (rw = '1') then -- если чтение
                       if (n bit = 7) then -- если прочитан байт
                   -- если данные не готовы – прекратить чтение
                        if new data = '0' then next state <= nack;
                        else -- иначе, если прочитаны все байты
                          if (num read = ALL READ REGS) then
                          next state <= <coстояние>;
                          else next state <= <состояние>:
                          end if: -- if num read = ...
                        end if: -- if new data = ...
                       -- иначе продолжаем чтение
                       else next state <= data tx rx;
                       end if: -- if (n bit = 7)
                      else -- если запись
```

**if** (n bit = 7) then -- если записан байт

```
next state <= <cостояние>;
                               -- иначе продолжаем запись
                               else next state <= data tx rx:
                              end if: -- if n bit = ...
                            end if: -- if rw = ...
                 when others => null:
                 end case:
       end process:
       -- процесс формирования сигналов на линиях SDA и SCL (gen scl)
      process (clock, state, ack, n bit, num read, num init, free,
new data, rw)
       variable buf 8: std logic vector(7 downto 0);
       begin
        if clock'event and clock = '1' then
          case state is
          when sleep \Rightarrow err \leq '0'; cnt \leq '0';
                           gen scl <= <лог. уровень>;
                           sda <= <лог. уровень>;
          when start \Rightarrow err \leq '0': cnt \leq '0':
                          gen scl <= <лог. уровень>;
                          sda <= <лог. уровень>;
          when slave addr =>
             buf 8 := conv std logic vector(<адрес Slave>, 7) & rw;
             err <= '0'; cnt <= '1';
             gen scl <= <лог. уровень>;
             -- для передачи лог. нуля устанавливать на линии SDA '0',
             -- для передачи лог. '1' - Z-состояние
             if (buf 8(7-n \text{ bit}) = '0') then sda \le \sqrt{3} ог. уровень>:
             else sda <= <лог. уровень>;
             end if:
          when sak =>
                          err <= '0'; cnt <= '0';
                          sda <= <лог. уровень>;
                          gen scl <= <лог. уровень>;
                          err <= '0': cnt <= '0':
          when mak =>
                          sda <= <лог. уровень>;
                          gen scl <= <лог. уровень>;
          when RS =>
                          err <= '0'; cnt <= '0';
                          sda <= RStart:
                          gen scl \leq <лог. уровень>;
```

```
when nack => err <= '0': cnt <= '0':
                   sda <= <лог. vровень>:
                   gen scl <= <лог. уровень>;
                   err <= '0': cnt <= '0':
   when stop =>
                   gen scl <= <лог. уровень>;
                   sda <= '0':
   when error \Rightarrow err \leq '1': cnt \leq '0':
                   sda <= 'Z'; gen scl <= '0';
   when data tx rx =>
          err <= '0': cnt <= '1':
          -- если началась запись данных ...
          if (rw = '0') and n bit = 0) then
          -- ... но шина SCL занята
           if (free = '0') then
             gen scl <= <лог. уровень>;
           else gen scl <= <лог. уровень>;
           end if: -- if free = ...
          end if: -- if rw = ...
          -- если выполняется чтение данных
          if (rw = '1') then
            sda <= <лог. уровень>;
          -- иначе, если выполняется запись
          -- если передается адрес регистра Slave SUB
           if data adress = '0' then
             buf 8 := conv std logic vector(addreses(num init),8);
           else -- иначе, если передается байт данных DATA
             buf 8 := conv std logic vector(words(num init),8);
           end if; -- if data adress = ...
           -- установка уровня на линии SDA для передачи '0' и '1'
           if (buf 8 (7-n bit) = '0') then sda \leq <\pi or. ypobehb>;
           else sda \le <\piог. уровень>;
           end if: -- if (buf 8(7-n \text{ bit}) = ...
          end if: -- if rw = ...
   when others => null;
   end case;
end if; -- if clock'event and clock = ...
end process;
-- счетчик бит со счетом по заднему фронту clk
-- и сбросом по cnt или free
process (clk, cnt, free)
 <операторы для счетчика бит>
```

```
-- анализ состояния линии SCL
process (clk. state)
begin
 if clk'event and clk = '1' then
  case state is
 -- анализ состояния SCL перед началом передачи данных
    when data tx rx => if n bit = 0 then free \leq= scl;
                        else free <= '1': end if:
 -- анализ линии SCL перед формированием условия P
   when stop \Rightarrow free \le scl:
  when others => free <= '1':
  end case:
 end if:
end process;
-- процесс формирования управляющих сигналов
process (clk, new data, num init, rw, init, type tx)
variable cycle: natural := 0; -- переменная для счета тактов clk
variable buf: data regs; -- буферный регистр
begin
 if (clk'event and clk = '1') then
  case state is
   when start \Rightarrow wd \leq '0': rw \leq '0':
                   -- если проинициализированы не все регистры
                   if (num init < ALL INIT REGS) then init <= '1';
                   else init \leq '0': end if:
                   ack <= <линия шины I2C>; RStart <= '1';
  when sak =>
                   -- определение типа следующих за АСК данных
                   if type tx = tx data then
                    if (init = '1') then
                     if (data adress = '0') then data adress <= '1':
                     else type tx \le tx stop;
                     end if: -- if data adress = ...
                    else
                     type tx \le tx rs;
                    end if: -- if init = ...
                   end if: -- if type tx =
  when slave addr => type tx <= tx adr; data adress <= '0';
  when data tx rx => if rw = '1' then
                            buf(num read)(7-n bit) := sda;
                            new data \leq= buf(0)(\leqномер бита\geq);
                        end if:
                         type tx \le tx data;
```

```
when sleep \Rightarrow rw \leq '0':
                   for i in 0 to ALL READ REGS-1 loop
                     <обнуление массива векторов buf(i)>:
                   end loop:
                   -- формирование сигнала начала нового цикла
                   if cvcle = 0 then wd \le '1'; new data \le '0';
                   else wd \le 0': end if:
                   num read <= 0; RStart <= '1';
                   if (new data = '1') then
   when nack =>
                     data out <= <выходные данные>:
                   end if:
   when mak => num read <= num read + 1;
   when RS \Rightarrow rw \Leftarrow '1'; RStart \Leftarrow '0';
   when stop \Rightarrow if init = '1' then num init \leq num init + 1;
                   end if:
   when others \Rightarrow null:
 end case:
-- счетчик тактов clk для задания интервала дискретизации
 if cycle = <значение> then cycle := 0;
 else cycle := cycle + 1;
 end if:
end if; -- if clk'event and clk = ...
end process;
end Behavior;
```

- 1.8. Задать настроечную константу делителя равной 8, константу DISCRETE = 200. Откомпилировать проект; при наличии в тексте программы ошибок внести исправления.
- 1.9. Выполнить моделирование работы проекта. Для упрощения отладки добавить в \*.vwf-файл помимо входных и выходных сигналов сущности текущее состояние ЦКА, служебные сигналы и переменную cycle. Задать частоту сигнала с кварца 24 МГц, время моделирования 200 мкс. По умолчанию установить на линии SCL логическую '1'. Запустить моделирование. Убедиться в правильности передачи байта с адресом Slave SAD по линии SDA.
- 1.10. Выполнить перезапись \*.vwf-файла результатами моделирования (П.1). Проанализировать правильность установки на линии SDA байта SUB или DATA.
- 1.11. Сымитировать в 9-м бите ответ АСК. Выполнить повторное моделирование.
- 1.12. Повторяя пп. 1.10 и 1.11, достигнуть состояния ЦКА data\_tx\_rx, в котором выполняется чтение данных из регистра

INT\_SOURCE, и сымитировать на линии SDA признак готовности данных.

- 1.13. Повторить моделирование. Убедиться в завершении цикла обмена сигналами NACK и P.
- 1.14. Сымитировать в состояниях ненулевой ответ от *Slave* в состояниях ЦКА data\_tx\_rx, в которых принимаются данные от устройства, и выполнить повторное моделирование.

### 2. Проверка работы проекта с датчиком ADXL345

- 2.1. Установить значение коэффициента делителя таким, чтобы обеспечить скорость обмена по шине I2C 400 кбит/с; изменить константу DISCRETE.
- 2.2. Подключить входные и выходные сигналы проекта к выводам ПЛИС отладочного комплекта. Линиям шины I2C, а также выходным портам сs и sad0 назначить номера выводов ПЛИС в соответствии со схемой подключения датчика ADXL345 к контактам разъема JP 1 (GPIO 0) или JP2 (GPIO 1). Выходной сигнал подключить к красным и зеленым светодиодам LEDR7 LEDR0 и LEDG7 LEDG0, порт егг к светодиоду LEDR9. Откомпилировать проект заново.
- 2.3. Не подключая плату с ПЛИС к ПЭВМ, соединить контакты разъема датчика с контактами разъема ЈР 1 (GPIO 0): VCC с контактом 3,3 В, GND с контактом GND. Показать преподавателю результаты подключения выводов датчика. Предельное напряжение обратной полярности для ADXL345 составляет всего -0,3 В, поэтому при неправильной коммутации выводов VCC и GND датчик будет поврежден.
  - 2.4. Включить отладочный макет. Загрузить проект в ПЛИС.
  - 2.5. Наблюдать на светодиодах изменение сигнала ускорения.
  - 2.6. Повторить п. 3 лабораторной работы №1.

## Лабораторная работа № 3

## Изучение протоколов передачи видеоданных VGA, DVI и LVDS

**Цель работы:** синтез устройства для реализации передачи видеоданных по протоколам VGA и LVDS.

#### ТЕОРЕТИЧЕСКАЯ ЧАСТЬ

Стандарт VGA (от англ. *Video Graphics Array* — массив видеографики, читается как «ви-джи-эй») — это аналоговый стандарт передачи видеоданных, который определяет размер кадра изображения, количество цветов и частоту обновления кадров; принят в 1987 г. Шина VGA содержит, как минимум, три аналоговых линии R, G и B для передачи соответственно сигналов красного, зеленого и

синего с уровнем напряжения от 0 до 0,7 В и две цифровых линии для передачи импульсов строчной HSYNC (*Horizontal synchronization*) и кадровой VSYNC (*Vertical synchronization*) синхронизации.

При поступлении импульсов синхронизации луч в электроннолучевой трубке монитора выполняет построчное сканирование, начиная с левого верхнего угла (на рис. 16 показаны номера строк и столбцов позиций луча для кадра размером 640х480 пикселей). При достижении конца строки луч переходит на строку ниже, а при достижении правого нижнего угла — снова в левый верхний угол. По окончании строки и кадра на линиях HSYNC и VSYNC формируются синхроимпульсы низкого логического уровня, переносящие луч

соответственно на начало строки и на нулевую строку. Также на этих линиях быть должны предусмотрены передний (Front Porch, FP) и задний (Back Porch, BP) защитные интервалы высокого логического уровня (верхняя эпюра на рис. 17).

При реализации на ПЛИС удобно начинать формирование синхросигнала не с BP, а с сигнала разрешения передачи видео (нижняя эпюра на рис. 17).

Основная (пиксельная) тактовая частота clk, выраженные в количестве ее периодов длительности логических уровней в линии



HSYNC и выраженные в количестве периодов HSYNC длительности в линии VSYNC строго регламентированы для каждого видеоформата.

Рассмотрим в качестве примера сигналы для формирования изображения 640x480 пикселей с частотой 60 Гц. Частота тактового сигнала clk для этого режима составляет 25,175 МГц, а длительности уровней в линиях синхронизации, выраженные в периодах clk и HSYNC, приведены в табл. 8.

Таблица 8

| Синхросигнал | Передача видео | FP | SYNC | BP | Всего |
|--------------|----------------|----|------|----|-------|
| HSYNC        | 640            | 16 | 96   | 48 | 800   |
| VSYNC        | 480            | 10 | 2    | 33 | 525   |

Стандарт DVI (от англ. Digital Visual Interface – цифровой видеоинтерфейс, читается как «ди-ви-ай») – цифровой стандарт который передачи видеоданных, определяет размер кадра изображения, количество цветов и частоту обновления кадров; принят в 1999 г. DVI использует высокоскоростную технологию TMDS Minimized Differential Signaling (Transition передача лифференциальных сигналов с минимизацией количества перепалов уровней): принцип заключается R таком кодировании последовательно передаваемых 10-битных слов. при котором количество переключений '0'->'1' и '1'->'0' в слове минимально.



При передаче видеоданных по одноканальной (Single Link) шине TDMS входные данные – 24 бита с информацией о цвете (по 8 бит на каналы RED, GRN и BLU), 4 бинарных управляющих сигнала импульсы И **HSYNC** VSYNC (рис. 18) - кодируются в Encoder Serializer блоке (сдвиговый регистр-кодировщик) 10-разрядный кол. который последовательно передается трем линиям связи типа «витая пара» (линии TX0-2) синхронизацией тактовым сигналом CLK (линия TXC). Алгоритм кодирования 10-битных слов определяется сигналом DE (**D**ata enable). Ha приемной стороне декодер выполняет обратную операцию, извлекая из

последовательных кодов сигналы R, G, B, CTL, DE, HSYNC и VSYNC. В передатчике используется умножитель частоты, повышающий частоту тактового сигнала пиксельной частоты ( $Pixel\ Clock$ )  $CLK_{PC}\ в$  10 раз: полученный тактовый сигнал CLK используется для тактирования сдвиговых регистров, а также передается по дифференциальной линии TXC.

Алгоритмы кодирования и декодирования для передачи сигналов по DVI рассмотрены в стандарте [7]. Логический уровень на линиях CTL определяет алгоритм кодирования при низком логическом уровне DE (в одноканальной шине TDMS на линиях CTL, как правило, устанавливается логический ноль).

Пример VHDL реализации передатчика DVI в соответствии с [7] приводится в интернет-источнике [8].

Стандарт LVDS (от англ. Low-Voltage Differential Signaling – передача низковольтного дифференциального сигнала, читается как «эл-вэ-дэ-эс») цифровой стандарт передачи видеоданных. разработанный компанией National Semiconductor в 1994 г. так же, как и DVI, реализует передачу высокочастотного дифференциального низковольтного (0,3 В) видеосигнала по линиям типа «витая пара» (шина TDMS). Исходный поток параллельных данных (24 или 18 бит цвета на пиксель), а также сигналы DE, HSYNC (HS) и VSYNC (VS) 7-битные конвертируются В слова. которые последовательно передаются по дифференциальным линиям ТХ. Тактовая частота, передаваемая по отдельной дифференциальной паре ТХС, получается умножением исходной частоты СLК РС в семь раз. Логика кодирования для передачи 24 и 18 бит на цвет приведена на рис. 19 и 20 соответственно.



#### Домашнее задание

- 1. Изучить основы передачи данных по протоколам VGA и LVDS.
- 2. Составить описание сущностей и архитектур цифровых устройств, синтезируемых в пп. 1 и 2 порядка выполнения лабораторной работы.

### ПОРЯДОК ВЫПОЛНЕНИЯ РАБОТЫ

- 1. Исследование передачи видеоданных по стандарту VGA
- 1.1. Создать новый проект с именем VGA\_Timer.
- 1.2. Для соответствующих номеру бригады параметров видеопотока (табл. 9) определить по [9] временные интервалы для импульсов HSYNC и VSYNC.

| гаолица 9. параметры видеопотока |       |       |      |       |      |      |  |  |  |
|----------------------------------|-------|-------|------|-------|------|------|--|--|--|
| №                                | 1     | 2     | 3    | 4     | 5    | 6    |  |  |  |
| Размер                           | 1024x | 1152x | 800x | 1024x | 800x | 800x |  |  |  |
| кадра, пикс.                     | 768   | 864   | 600  | 768   | 600  | 600  |  |  |  |
| Частота. Ги                      | 60    | 75    | 100  | 75    | 75   | 85   |  |  |  |

Таблица 9. Параметры видеопотока

1.3. Подключить к проекту пакет с объявлением типа VGA\_params, представляющего собой запись с полями Video, FP, Sync и BP, а также констант H\_TIMING и V\_TIMING данного типа, указав в данных полях моменты времени от начала периода сигнала синхронизации: например, для видеосигнала 640x480 пикселей с частотой кадров 60  $\Gamma$ ц в соответствии с [9]:

- 1.4. Объявить сущность с настроечными константами H и V типа VGA\_рагат и значениями по умолчанию H\_TIMING и V\_TIMING соответственно, входным портом clk, выходными портами hsync и vsync, портами гоw и column типа **natural** ограниченной разрядности для обозначения номера строки и столбца пикселя соответственно и логическим портом video\_on для разрешения выдачи сигнала в линии R, G и B.
- 1.5. Написать содержательную часть архитектуры. Сформировать сигналы hsync и vsync с требуемыми временными интервалами для формата видеоизображения в соответствии с номером варианта. Сформировать высокий логический уровень video\_on в моменты времени, соответствующие передаче видео, и низким в противном случае. Сформировать сигналы с номером столбца и строки.
- 1.6. Выполнить моделирование работы проекта для частоты  $f_{\rm clk} = 50~{\rm M}\Gamma$ ц на интервале времени 3-5 периодов hsync, убедиться в его работоспособности.
- 1.7. Создать новый проект с именем VGA\_Video\_Generator с входными портами clk, row, column и video\_on и 4-разрядными выходами R, G и B типа **std\_logic\_vector**.
- 1.8. Написать содержательную часть архитектуры со следующей логикой работы. При video\_on = '0' значения R, G и B асинхронно обнуляются. При video\_on = '1' по переднему фронту clk генерируются коды R, G и B для формирования тестового кадра из 8 цветных вертикальных полос одинаковой ширины с цветами (слева направо): белый, желтый, голубой, зеленый, пурпурный, красный, синий, черный. Интенсивность сигналов (кроме черного): 75% от максимального значения.

- 1.9. Выполнить моделирование работы проекта для первой строки, убедиться в его работоспособности.
- 1.10. Создать новый проект с именем VGA с входным тактовым сигналом clock и выходными портами hsync, vsync, R, G и B.
- $1.11.~\mathrm{B}$  декларативной части подключить к проекту компонент аппаратно реализованного генератора с петлей ФАПЧ (PLL, от англ. *Phase Locked Loop*) для формирования тактового сигнала требуемой пиксельной частоты из сигнала кварцев 24, 27 или 50 МГц (**П.2**). Принцип работы такого генератора подробно описан в [10]. Также подключить компоненты синтезированных в пп.  $1.1~\mathrm{u}$  1.7 проектов.
- 1.12. С использованием структурного стиля программмирования разработать содержательную часть архитектуры в соответствии с рис. 21.
- 1.13. Выполнить моделирование работы проекта для первой строки, убедиться в его работоспособности.



- 1.14. Подключить входные и выходные сигналы проекта к соответствующим выводам ПЛИС отладочного комплекта, воспользовавшись описанием к лабораторному макету *DE1 User Manual*.
  - 1.15. Подключить монитор к разъему DB-15F отладочного макета.
- 1.16. Загрузить проект в ПЛИС, проверить его работоспособность.

## 2. Исследование передачи видеоданных по стандарту LVDS

- 2.1. Изменить константы H\_TIMING и V\_TIMING проекта VGA\_Timing на значения для формата видео 800х600 пикселей с частотой кадров 60 Гц [9]. Изменить разрядность выходных сигналов сущности VGA\_Video\_Generator на 8.
- 2.2. Создать новый проект с именем LVDS\_coder с входами clk, hsync, vsync и de типа **std\_logic**, 8-разрядными входами R, G и B и пятью логическими выходами TX0-TX3 и TXC для кодирования согласно рис. 18. Разработать содержательную часть его архитектуры.
- 2.3. Создать новый проект LVDS передатчика тестового сигнала с именем LVDS\_tx с входом clock и выходами TX0-TX3 и TXC. Подключить к проекту компоненты VGA\_Timing, VGA\_Video\_Generator и LVDS\_coder, а также компонент (генератор с петлей ФАПЧ) для повышения частоты clk в 7 раз.



2.4. Составить архитектуру для реализации передатчика LVDS в соответствии с рис. 22.

2.5. Выполнить моделирование работы проекта для  $f_{clock} = 50 \text{ MFц}$  на интервале 3-4 периодов hsync. При необходимости вынести в файл моделирования вспомогательные сигналы.

Доказать правильность работы синтезированной сущности.

### Лабораторная работа № 4 Изучение основ проектирования систем на кристалле на примере процессора Nios II

**Цель работы:** синтез микропроцессорной (МП) системы с использованием ядра Nios II.

конце 90-x 2000-x начале годов на смену специализированным микросхемам пришли микроэлектронные разработки, основанные на программных моделях ядер устройств. Описания ядер на языках группы HDL позволили компоновать проекты из готовых и хорошо отлаженных модулей. Составленный таким образом проект принято называть «системой на кристалле» (от англ. «System-on-crystal», SoC). Синтезируемые МП системы можно на ПЛИС типа FPGA: фирма Xilinx предлагает реализовать процессорные ядра Microblaze и Picoblaze (32-х и 8-разрядные соответственно), фирма Altera – ядро Nios II.

Nios II — это встраиваемый программный 32-разрядный процессор общего назначения с перестраиваемой конфигурацией, оптимизированный для реализации на ПЛИС компании Altera со структурой FPGA. Nios II построен по гарвардской архитектуре, т.е. имеет две раздельные шины адреса и данных; разрядность каждой шины составляет 32 разряда. В его архитектуру также входят 32 регистра общего назначения (РОН) и 32 источника внешних прерываний. Допустимый объем внешнего адресного пространства — 2 Гб. Nios II является процессором конвейерного типа (т.е. команды в нем выполняются за один цикл частоты синхронизации), использует сокращенную систему команд (RISC) и поддерживает вычисления в формате с плавающей точкой. В процессорах RISC арифметические и логические операции выполняются над операндами, находящимися в

регистрах общего назначения. Обмен информацией между регистрами и памятью МП системы осуществляется путем выполнения команд загрузки данных из регистра и сохранения данных в регистр.

Структура процессора Nios II показана на рис. 23.



Рис 23

Блоки памяти команд и данных, помеченные как TC (от англ. *Tightly Coupled* – тесно связанные), принадлежат специальному типу памяти, обладающему малыми задержками и применяемому для

критичных к быстродействию приложений. Такая память реализуется на кристалле, но вне ядра [11, C.607-620]. Так называемые теневые регистры (от англ. *shadow registers*) программисту не доступны – с ними может работать только МП. Типовое использование теневых регистров состоит в ускорении контекстных переключений.

Блок опережающей выборки команд необходим для конвейеризации — ускорения выполнения команд МП. Выполнение любой команды условно можно разделить на два этапа: выборку кода команды из памяти и непосредственно ее выполнение. Конвейеризация позволяет сократить до минимума первый из этапов: при этом выборка команды из памяти и ее дешифрирование в стандартную для АЛУ форму объединяются так, что в любой момент времени несколько шагов выполняются одновременно, т.е. одна команда выбирается из памяти, вторая дешифрируется, а третья исполняется.

Наличие одновременно внутреннего и внешнего контроллеров прерываний не является противоречием, так как при работе внешнего контроллера внутренний отключается.



Рис. 24

Архитектура Nios II поддерживает до определяемых пользователем называемых заказных, от англ. «custom logic») команд. АЛУ Nios II напрямую соединяется с логикой заказной команды, позволяя аппаратно реализовать операции, обращение к которым и использование которых происходит по аналогии стандартными командами. Использование заказных команд позволяет добавить к АЛУ аппаратный узел собственной разработки, обрабатывающий за один задачи несколько тактов, т.е. заменить сложную последовательность стандартных МΠ единственной командой, которая

аппаратными средствами выполняет те же действия. Добавленная пользователем заказная логика также может обращаться к памяти и логике вне Nios (рис. 24). Здесь необходимо отметить основное отличие МП, реализованного на ПЛИС, от стандартного МП. В стандартном МП скорость обработки зависит только от скорости работы самого микропроцессора (тактовой частоты), ресурсов (разрядность шин, число регистров, наличие умножителей и т. д.) и эффективности программного кода. При этом все ресурсы МП, как правило, участвуют в решении задачи последовательно, а данные из

одного вычислительного процесса переносятся другой вычислительный процесс под управлением программы. Для МП, реализованного на ПЛИС, ресурсами являются логические ячейки самой ПЛИС, и формально не важно, как именно данные ресурсы связаны с МП – как отдельные вычислительные узлы (например, умножители с накоплением) или как конечные автоматы. После того пользователь определил ресурс, требуемый ДЛЯ специфичных аппаратных задач, все остальные ресурсы ПЛИС могут быть задействованы для реализации вычислительных узлов. Также при реализации вычислений на заказной логике данные передаются из одного вычислительного процесса в другой аппаратно, не требуя затрат времени МП [12].

К собственно процессорному ядру в процессе синтеза МП системы возможно подключение дополнительных ядер — *Intellectual Property cores (IP-cores)*: таймеров, блоков постоянной и оперативной памяти, модулей аппаратно реализованных интерфейсов (Ethernet, UART, SPI и др.), широтно-импульсных модуляторов и т.п. либо ядер, написанных пользователем.

Для Nios II доступны три конфигурации МП ядра: экономичная Nios II/е (*economy*), стандартная /s (*standart*) и быстрая /f (*fast*). Выбор конкретной конфигурации определяется ресурсами ПЛИС, сложностью вычислений и требуемым набором средств отладки. На реализацию перечисленных разновидностей ядра требуется соответственно не более 600, 1300 и 1800 логических элементов ПЛИС. Производительность МП в микросхемах семейства Stratix достигает для перечисленных вариантов 0,16, 0,75 и 1,17 DMIPS/МГц, что при работе на максимальных частотах ядер (135...150 МГц) соответствует 28, 120 и 200 DMIPS. Dhrystone MIPS, или DMIPS, — синтетический тест производительности МП; MIPS — количество миллионов инструкций в секунду (от англ. *million instructions per second*).

Подключение к процессору внешних устройств осуществляется либо с помощью раздельных шин данных для записи и для чтения, либо с использованием шины с тремя состояниями.

Для построения системы на кристалле (СнК) с процессором Nios II используется шина Avalon. Ее основными отличительными особенностями являются простота, малое количество ячеек кристалла, задействованных для логики шины, и полностью синхронные операции.

Каждая шина Avalon подключает единственного *Master* к нескольким периферийным устройствам. *Master* шины Avalon обычно выступает ядро микропроцессора Nios II. Периферийным устройством Avalon может быть, например, UART, таймер, или устройство памяти.

Все транзакции шины Avalon передают (перемещают) единственный байт, слово половинной разрядности или слово (8, 16, или 32 бита) между одним из периферийных устройств и *Master*. Шина Avalon адаптирована для синтеза на кристалле: для взаимосвязи устройств вместо трехстабильных шин используются мультиплексоры.

Как правило, шина Avalon является модулем с большим количеством линий ввода-вывода. Модуль шины имеет один специальный периферийный порт интерфейса (peripheral special interface port, PSIP) для каждого Slave и один PSIP для Master («владельца» шины), а также содержит логику, которая выполняет следующие функции:

декодирование адреса – производит сигналы выбора кристалла для каждого периферийного устройства;

мультиплексирование шины данных — передает данные от выбранного  $Slave \ \kappa \ Master;$ 

динамическое изменение разрядности шины – для чтения или записи данных большей разрядности, чем у периферийного устройства, автоматически выполняются многократные циклы шины;

назначение номера прерывания – определяет номер и приоритет прерывания IRQ (*Interrupt request*), когда один или более *Slave* одновременно требуют запроса прерывания.

При добавлении либо удалении *Slave* вся связанная с ним информация (PSIP) также добавляется или удаляется из шины Avalon.

Процессор Nios II поддерживает полный набор встроенных системных инструментальных средств развития: C/C++ транслятор, ассемблер и JTAG-отладчик. Ядро Nios II (в зависимости от конфигурации) включает поддержку аппаратных контрольных точек и управления через интерфейс JTAG.

Наиболее полное руководство по работе с процессором Nios II (на английском языке) приводится на сайте разработчика [13]. Для выполнения лабораторной работы можно воспользоваться справочными материалами из пособий [14, 15].

#### Домашнее задание

- 1. Изучить основы архитектуры МП Nios II.
- 2. Воспользовавшись программой *WinFilter.exe*, определить в соответствии с табл. 10 весовые коэффициенты фильтра авторегрессии скользящего среднего (APCC) для фильтров нижних (ФНЧ) и верхних (ФВЧ) частот и порядка фильтра n=3.

| гаолица 10. Тип цифрового фильтра и частота среза |          |     |      |         |     |          |  |
|---------------------------------------------------|----------|-----|------|---------|-----|----------|--|
| №                                                 | 1        | 2   | 3    | 4       | 5   | 6        |  |
| Тип фильтра                                       | ФВЧ      |     |      | ФНЧ     |     |          |  |
| Тип АЧХ                                           | Чебышева |     | Бесо | Бесселя |     | Чебышева |  |
| Относительная частота среза                       | 0,15     | 0,2 | 0,25 | 0,25    | 0,2 | 0,15     |  |

- 3. Составить характеристическое уравнение АРСС фильтра и построить его амплитудно-частотную характеристику (АЧХ) с помощью Mathcad.
- 4. Построить переходную характеристику (ПХ) фильтра для числа отсчетов не менее 5. Перевести отсчеты ПХ фильтра сначала в дополнительный 8-разрядный код в формате с фиксированной точкой (старший разряд – знаковый, далее фиксированная точка, далее 7 информационных бит дробной части), а затем – в шестнадцатеричный кол.
- 5. Составить описание программы на языке С для фильтрации цифровых отсчетов в соответствии с характеристическим уравнением АРСС фильтра.

### ПОРЯДОК ВЫПОЛНЕНИЯ РАБОТЫ

### 1. Синтез структуры МП системы (аппаратная часть)

- 1.1. Создать новый проект с именем Nios proc. В меню *Tools* выбрать мастер настройки системы на кристалле Osvs. В результате будет создан новый модуль СнК - генератор тактового сигнала и сигнала сброса (Clock Source) с именем clk 0. Переименовать модуль в clk, вызвав контекстное меню на его имени либо нажав F2. Сохранить СнК с именем nios сри.
- 1.2. Дважды нажать левой кнопкой мыши (ЛКМ) на модуле Clock Source и в открывшемся диалоговом окне установить частоту тактового сигнала 50 МГц (50000000 Гц) в поле Clock frequency. Нажать *Finish*, чтобы применить изменения.
- 1.3. Добавить в СнК модуль МП ядра Nios II из библиотеки (Library) Processors двойным нажатием ЛКМ на Nios II Processor либо кнопку Add (добавить). Выбрать минимальную конфигурацию МП (Nios II/e) в диалоговом окне. Поскольку карты памяти МП системы еще не сформирована, то дальнейшее конфигурирование модуля МП временно не производить. На странице JTAG Debug Module (JTAGотладчик) выбрать опцию Level 2 для поддержки основных отладочных функций (загрузка программы по интерфейсу JTAG, программные прерывания). Нажать *Finish*. Переименовать модуль МП в nios сри.
  - 1.4. Добавить модуль ОЗУ программ и данных: *Memories and*

Метоту Controllers -> On-Chip -> On-Chip Memory (RAM or ROM). В открывшемся диалоговом окне выбрать ОЗУ (RAM) и задать размер памяти *Total Memory Size* 8192 байта, размер слова данных *Data Width* − 32 (бита). Остальные параметры оставить без изменений. Нажать «Finish». Переименовать модуль в on chip RAM.

- 1.5. Аналогичным образом добавить модуль ПЗУ (ROM) и переименовать в on\_chip\_ROM.
- 1.6. Добавить компонент аппаратного идентификатора версии System ID Peripheral из библиотеки *Peripherals -> Debug and Performance -> System ID Peripheral*. Идентификатор обновляется при каждой генерации СнК и позволяет контролировать соответствие версий сгенерированного программного кода и СнК. Если значение идентификатора в программном обеспечении и в загруженной в ПЛИС СнК не совпадает, при загрузке программного обеспечения будет выдана ошибка. Задать имя модуля *sysid*, значение идентификатора *System ID* (в десятичном коде) ГГОN, где ГГ последние две цифры текущего года, N номер бригады.
- 1.7. Добавить модуль UART из библиотеки *Interface protocols -> Serial -> UART (RS-232 Serial Port)* и принять его параметры по умолчанию: количество бит данных *Data bits* 8, стоповый бит *Stop bits* 1, бит четности отсутствует (*Parity None*), скорость передачи данных *Baud Rate* 115200 бит/с.

Поскольку все добавленные модули пока не соединены между собой, в нижней части рабочего окна мастера Qsys будут отображены сообщения об ошибках при синтезе МП системы.

- 1.8. Создать сеть синхронизации и сброса СнК. Для этого на вкладке System Contents рабочего окна Qsys в столбце Connections соединить выходы блока синхронизации и сброса Clock Output (clk) и Reset Output (clk\_reset) с входами clk (clk1) и reset (reset1, reset\_n) модулей МП, памяти, идентификатора системы и UART соответственно. Соединение осуществляется нажатием ЛКМ на узле соединения линий (выполненное соединение отображается заливкой черным цветом).
- $1.9.\$ Подключить все блоки МП СнК к системной шине Avalon: выходы Data master и Instruction Master процессора NIOSII/е подключить к входу s1 модулей ОЗУ, ПЗУ и UART, а также входу  $control\ slave$  идентификатора системы.
- 1.10. Экспортировать сигналы tx и rx модуля UART во внешние порты. Для этого во вкладке  $System\ Contents$  выбрать компонент UART и сигнал  $external\ connection$ , затем, нажав ЛКМ в столбце Export, указать название сигнала uart и нажать Enter.
  - 1.11. Соединить выход запроса на прерывание IRQ блока UART

с входом IRQ процессора (IRQ 31) в столбце IRQ.

- 1.12. Выполнить настройку векторов исключений и прерываний МП (отличие между понятиями «исключение» и «прерывание» МП рассмотрено в **П.3**). Дважды нажать ЛКМ на модуле МП Nios II/е. В открывшемся диалоговом окне в поле Reset Vector (адрес ячейки памяти при сбросе) выбрать параметр Reset Vector Memory: on\_chip\_RAM.s1. В поле Exception Vector (адрес ячейки памяти при исключении) выбрать параметр Exception Vector Memory: on\_chip\_RAM.s1 (другие настройки МП кратко рассмотрены в **П.4**). Нажать кнопку Finish.
- 1.13. Назначить базовые адреса устройств МП системы и приоритет прерываний: команды *System -> Assign Base Addresses* и *System -> Assign Interrupt Numbers* соответственно.
- 1.14. На вкладке *HDL Example* в выпадающем списке *HDL Language* выбрать язык описания VHDL и скопировать в буфер обмена VHDL-описание компонента МП.
- 1.15. Сохранить конфигурацию МП СнК и закрыть окно мастера Qsys.
- 1.16. Создать новый VHDL-файл. Объявить в нем сущность nios\_proc с входами тактового сигнала и сброса, а также портами UART. Подключить файл с описанием МП nios\_cpu.qsys через меню **Project** -> **Add/Remove Files in Project**. Подключить к проекту компонент МП. Скомпилировать проект. Подключить входные и выходные сигналы проекта к выводам ПЛИС отладочного комплекта (для формирования сигнала сброса использовать одну из кнопок **Key**) и выполнить повторную компиляцию.

## 2. Разработка программной части

- 2.1. Запустить программу Nios II Software Build Tools for Eclipse (далее по тексту Eclipse) и в открывшемся диалоговом окне выбрать для сохранения проекта Eclipse каталог с проектом Nios proc.
- 2.2. Для создания нового проекта использовать меню *File -> New -> Nios II Application and BSP from Template*. В открывшемся диалоговом окне настроек в качестве шаблона проекта *Template* выбрать *Blank project* (пустой проект); на панели *Target Hardware information* (сведения об аппаратной части СнК) указать:

SOPC Information File Name: ... /nios\_cpu.sopcinfo; project name: nios\_cpu.

Для применения сделанных изменений нажать кнопки Next и Finish. В результате будут созданы два проекта: проект nios\_cpu\_bsp с набором драйверов, настроек и библиотек для аппаратной МП системы (Board support package, BSP) и шаблон проекта nios\_cpu.

**Примечание:** при работе с Eclipse на ПЭВМ с операционной системой Windows 7 и выше возможно появление сообщения об ошибке о невозможности создания проекта «Failed to execute: ./create-this-app --no-make». В этом случае нужно выполнить запуск Eclipse с правами администратора, вызвав в меню Пуск на ярлыке Eclipse контекстное меню и выбрав пункт «Запуск от имени администратора».

- 2.3. Добавить С-файл в проект. Для этого вызвать контекстное меню каталога nios\_cpu в дереве проектов *Project Explorer*, выбрать *New->Other->Source File*, нажать *Next*, в появившемся диалоговом окне в поле *Source file* ввести имя нового файла «nios cpu.c» и нажать *Finish*.
- 2.4. Добавить в файл nios\_cpu.c код программы приема данных по UART, их фильтрации и выдачи результата по UART.

```
// подключение заголовочного файла с описанием
// функций ввода/вывода данных
#include "sys/alt stdio.h"
// основная программа
int main()
 unsigned char TxRx; // байт данных в доп. коде
 // переменные для цифрового фильтра
 // с присвоением нулевых значений по умолчанию
 float x[<константа 1>], y[<константа 2>];
 int i; // счетная переменная для организации цикла
 while (1)
   {
    // чтение одного байта данных из порта UART
    TxRx = alt getchar():
    // перевод из дополнительного кода в число со знаком
    // в формате с фиксированной точкой
    x[0] = (TxRx < 128)? TxRx/128.0 : -1*(256 - TxRx)/128.0;
    <операторы для реализации цифрового фильтра>;
    // перевод числа со знаком из формата с
    // фиксированной точкой в дополнительный код
    TxRx = (y[0] > 0)? ((unsigned char)128*y[0]) :
                              (256 + (unsigned char)(128*y[0]));
     // запись одного байта данных в порт UART
    alt putchar(TxRx); }
return 0;
Сохранить файл с кодом программы.
```

2.5. Создать образ слоя абстрагирования (Hardware Abstraction Layer, HAL), позволяющий взаимодействовать инструкциям высокоуровневого языка программирования С/С++ с низкоуровневыми компонентами (аппаратным обеспечением) и содержащий драйверы устройств, используемых в аппаратном проекте. Для этого в окне Есlipse выбрать пункт Nios II -> BSP Editor, в открывшемся диалоговом окне выбрать меню File -> Open и указать путь к файлу настроек СнК settings.bsp (...\Software\nios\_cpu\_bsp\). Оптимизировать НАL для экономии объема памяти, занимаемого программой (П.5).

Нажать на кнопку *Generate* в окне Eclipse. Сохранить сгенерированный слой абстрагирования. Закрыть окно *BSP Editor*.

- 2.6. Выполнить сборку программного проекта (команда **Project -> Build All**). При необходимости исправить синтаксические ошибки в тексте программы.
- 2.7. Оценить объем памяти, занимаемой программой: в окне дерева проектов выбрать файл *nios\_cpu -> Binaries -> nios\_cpu.elf* и в поле служебной информации на вкладке *Console* найти строку

Info: (nios\_cpu.elf) *nnn* Bytes program size (code + initialized data). Файл с расширением .elf представляет собой программу для загрузки в МП Nios II (**П.6**).

- 3. Загрузка программы в синтезированный процессор и ее отладка
- 3.1. Подключить кабель RS-232 к разъемам отладочного макета и ПЭВМ. Загрузить аппаратный проект СнК nios\_proc в ПЛИС средствами Quartus II.
- 3.2. Выполнить загрузку программного проекта МП nios\_cpu в ПЛИС средствами Eclipse. Выбрать пункт меню *Run -> Debug Configurations...* Создать новую отладочную сессию *Nios II Hardware ->New Configuration*. В поле *Project Name* выбрать nios\_cpu. Убедиться в активности соединения с микросхемой загрузчика USB-Blaster на вкладке *Target Connection* (в таблице *Processors* должна быть указана строка с процессором Nios II). Отметить пункты *«Ignore mismatched system timestamp»* (игнорировать время сборки МП системы) и *«Reset the selected target system»* (после загрузки выполнить сброс МП системы). Нажать кнопку *Debug*.
- 3.4. После загрузки программы в процессор выполнить анализ работы проекта с использованием программы терминала *HTerm.exe*, позволяющей принимать и передавать данные через СОМ-порт ПЭВМ. После запуска *HTerm.exe* подключиться к активному СОМ-порту ПЭВМ (как правило, это порт Com1) и настроить параметры передачи в соответствии с настройками модуля uart CнК (П.7).

- 3.5. Подать на вход процессора функцию единичного скачка, записав максимально допустимое для рассчитанного цифрового фильтра положительное значение (в кодировке hex) в строке поля Input control (П.7).
- 3.6. Проанализировать ответ МП Nios II, сравнив результаты домашних расчетов и первый байт принятых данных ПХ.
- 3.7. Повторить пп. 3.5 и 3.6 для получения остальных отсчетов ПХ цифрового фильтра.
- 3.8. Проанализировав принятые от МП байты данных со значениями отсчетов ПХ, сделать вывод о правильности работы цифрового фильтра. При наличии ошибок использовать точки останова в Eclipse и пошаговую отладку (П.8) для просмотра текущих значений переменных в функции main().
- 3.9. Изменить код программного проекта: закомментировать в нем строки, реализующие цифровой фильтр, и добавить строки, реализующие передачу по UART значения идентификатора системы (System ID) в виде четырех байт. Для этого использовать файлы описаний, находящиеся в BSP проекте:

#include "system.h"

#include "altera avalon sysid qsys.h"

#include "altera avalon sysid qsys regs.h"

#include "alt\_types.h"

Для получения System ID использовать функцию

IORD\_ALTERA\_AVALON\_SYSID\_QSYS\_ID(SYSID\_QSYS\_0\_BASE), где SYSID\_QSYS\_0\_BASE – базовый адрес блока *sysid*, который автоматически записывается в файл system.h при сборке проекта. Для хранения идентификатора проекта ввести переменную

alt\_u32 hardware\_id;

Тип alt\_u32 – Altera unsigned 32 bit – аналог типа unsigned int языка С.

Для разделения hardware\_id на 4 байта использовать битовые маски и операцию побитового И.

3.10. Выполнить компиляцию программного проекта, загрузить его в ПЛИС; с помощью программы терминала принять значение идентификатора СнК *sysid* и сравнить его со значением, присвоенным в п. 1.6.

# Приложение

**П.1.** Для перезаписи файла моделирования необходимо на странице с результатами моделирования *Simulation Report* выбрать инструмент *Waveform Editing Tool*  $\stackrel{\mathcal{H}}{\to}$  и в появившемся окне предупреждения выбрать пункт «*I want to overwrite my vector input file* ...»

(«Я желаю перезаписать исходный файл моделирования ...»).

- П.2. Создание компонента ФАПЧ.
- 1. В меню *Tools* выбрать мастер подключения компонентов *MegaWizard Plug-In Manager*.
- 2 В открывшемся диалоговом окне выбрать пункт *Create a new custom megafunction variation* (создать новую мегафункцию).
- 3. В новом диалоговом окне выбрать язык описания VHDL; в списке доступных мегафункций в левой части окна раскрыть папку I/O, нажав на «+» слева от нее, и выбрать компонент ALTCLKLOCK; в поле «What name do you want for the file?» (какое имя Вы хотите назначить файлу?), ввести идентификатор (например, pll<выходная частота>МНz) и нажать кнопку Next. Появится многостраничное окно с настройками компонента ФАПЧ ALTCLKLOCK.
- 4. В окне **Parameter Settings: General/Modes** (основные настройки) в поле «**What is the frequency of the inclock0 input?**» (частота входного сигнала) ввести значение частоты выбранного кварца (24, 27 или 50 МГц).
- 5. В следующем окне *Parameter Settings: Scan/Inputs/Lock* (дополнительные входы) снять галочки в пунктах «*Create an 'areset' input ...*» (создать вход асинхронного сброса) и «*Create 'locked' output ...*» (создать блокируемый выход). Удаление указанных портов можно контролировать на условном графическом обозначении компонента в левом углу окна.
- 6. В окне «Output clocks: clko» выбрать вариант «Enter output clock frequency» (ввод частоты выходного сигнала) разблокированном поле ввести требуемое значение выходной частоты. Если данное значение  $f_1$  можно представить в виде  $f_1 = n_1 f_0 / n_2$ , где  $n_1$  и  $n_2$  – натуральные числа, меньшие 64, то в верхней части окна текстом синего цвета будет показано подтверждение «Able to implement ...» (возможно применить). Значения  $n_1$  и  $n_2$  при этом отобразятся в полях «Clock multiplication factor» (множитель) и «Clock division factor» (делитель) соответственно. В противном случае текстом красного цвета будет показано подтверждение «Cannot implement ...» (нельзя применить), т.е. компонент ФАПЧ с текущими настройками не может быть синтезирован. В этом случае необходимо вернуться к окну выбора входной частоты и изменить ее значение. Также в окне «Output clocks: clk0» можно задать скважность выходного сигнала (Clock duty cycle) и сдвиг его фазы (Clock phase shift).
- 7. При необходимости аналогичным п. 6 образом в полях «*Output clocks: clk1(2)*» можно задать еще две выходных частоты.
  - 8. В окне *Summary* (результаты) можно выбрать файлы, которые

будут созданы в директории проекта (обязательно должен быть выбран файл с расширением \*.vhd).

**П.3.** Прерывание и исключение – событие, означающее, что в МП системе и/или выполняющейся программе возникло состояние, требующее незамедлительной реакции на него процессора. Как правило, прерывания и исключения приводят к принудительной передаче управления от текущей программы к специальной программной процедуре, называемой обработчиком прерывания/исключения.

Прерывания возникают в случайные моменты времени в ответ на сигналы управления от внешних цифровых устройств или внутренних таймеров МП системы.

Исключения возникают в моменты времени, когда процессор определяет возникновение ошибки в процессе выполнения программы (деление на ноль, корень квадратный или логарифм от отрицательного аргумента и др.).

При появлении запроса на прерывание или возникновении исключения текущая задача приостанавливается на время, в течение которого МП выполняет код обработчика. Когда выполнение обработчика прекращается, МП возобновляет выполнение прерванной процедуры или задачи.

- **П.4.** На странице Caches and Memory Interfaces (кэш-память) в конфигурациях Nios II/s и Nios II/f пользователь может установить объем кэша команд (для конфигурации Nios II/e он по умолчанию равен 4 кбайт) и поддержку пакетных передач (Bursts Transfers) с настройкой объема пакета. В лабораторной работе память программ расположена во внутренней статической памяти МП, поэтому пакетную передачу включать не требуется (при использовании динамической памяти включение пакетных передач желательно). При использовании внешних модулей памяти (ОЗУ и ПЗУ) также рекомендуется задействовать один порт тесно связанной памяти команд (tightly coupled instruction master port), который используется для прямого доступа МП к модулям оперативной памяти на кристалле, что повысить производительность избежать конфликтов (например, И одновременном доступе к общей внешней шине микропроцессорной системы шин данных и инструкций процессорного ядра).
- **П.5.** Для оптимизации памяти на панели *hal* ветви *Common* (общие настройки) дерева настроек *Settings* оставить галочки только на пунктах *«enable\_reduced\_device\_drivers»* и *«enable\_small\_c\_library»* (разрешить сокращение списка драйверов и использование упрощенной библиотеки С). Поскольку программа для СнК в

лабораторной работе составлена на языке C и работает в бесконечном цикле, также можно отключить поддержку C++ и опцию разрешения выхода из программы. Для этого на панели **hal** дерева *Settings* снять галочки в пунктах *«enable\_c\_plus\_plus»* (поддержка C++), *«enable\_clean\_exit»* и *«enable\_exit»* (разрешение выхода).

Убедиться, что выбран пункт «enable lightweight device driver api».

**П.6.** Файл конфигурации .sof («прошивка», формируемая Quartus II) загружается в конфигурационное ОЗУ ПЛИС для формирования аппаратной части СнК, а файл .elf («прошивка», формируемая Eclipse) загружается в ОЗУ МП, чтобы программа Nios II запустилась с нужной точки (как правило, с адреса 0х00..0).

При автономной работе СнК файлы .sof и .elf загружаются не напрямую в ПЛИС, а сначала в конфигурационное ПЗУ (в этом случае адреса .sof и .elf файлов записываются со смещением) [16]. Чтобы .sof и .elf файлы в этом случае были размещены в конфигурационной памяти ПЛИС друг за другом, в СнК используется специальная микросхема – *EPCS Serial Flash* контроллер (флэш-память). Контроллер имеет в своем составе специальный загрузчик (boot loader) — программу, которая запускается при включении питания. Boot loader выполняет последовательную загрузку аппаратной и программных частей проекта и переводит вектор состояния МП на начало программы.

- **П.7.** В выпадающем списке *Port* окна программы *HTerm.exe* выбрать доступный СОМ-порт. В полях справа от списка портов задать скорость передачи (*Baud*), длину информационной части в битах (*Data*), количество стоповых бит в протоколе (*Stop*) и тип бита четности (*Parity*). Установить шестнадцатеричную кодировку передаваемых и принимаемых байт, выбрав *hex* в выпадающем списке *Type* панели *Input control*. Подключиться к выбранному СОМ-порту, нажав на кнопку *Connect*. Для отправки байта данных по UART ввести его значение в поле панели *Input control* и нажать кнопку *ASend*. В появившемся служебном окне в поле *Repetitions* задается количество повторов посылок байта (при вводе нулевого значения устанавливается режим непрерывных посылок). Отправленные в МП систему байты данных отображаются в поле *Transmitted data*, принятые из МП систему байты в поле *Received data*.
- **П.8.** Команды для пошаговой отладки доступны в меню *Run*. Точка останова ставится двойным нажатием ЛКМ в поле, расположенном слева от кода программы.

# Библиографический список

1. Новицкий А. Синхронный последовательный интерфейс SPI в микроконтроллерах «от А до Я» и его реализация на примере

ADuC70xx фирмы Analog Devices // Компоненты и технологии. – 2009. – № 3. – С. 53-60.

- 2. http://www.gaw.ru/html.cgi/txt/interface/spi/index.htm
- $3.\ http://www.analog.com/media/en/technical-documentation/data-sheets/ADXL345.pdf$
- 4. Семёнов Б.Ю. Шина I2С в радиотехнических конструкциях. М.: Солон-Р, 2002. 190 с.
  - 5. http://easyelectronics.ru/interface-bus-iic-i2c.html
  - 6. http://robocraft.ru/blog/communication/780.html
  - 7. Digital Visual Interface. Revision 1.0. 1999. 76 p.
  - 8. http://hamsterworks.co.nz/mediawiki/index.php/Dvid\_test
  - 9. http://tinyvga.com/vga-timing.html
- 10. Васильев Е.В. Цифровые радиопередающие устройства: учеб. пособие. Рязань: РГРТУ, 2006. 72 с.
- 11. Угрюмов Е.П. Цифровая схемотехника: учеб. пособие для вузов. 3-е изд., перераб. и доп. СПб.: БХВ-Петербург, 2010. 816 с.
- 12. Каршенбойм И. Микроконтроллер для встроенного применения NIOS. Система команд и команды, определяемые пользователем. Часть II. Команды перехода, исключения, конвейер и команды, определяемые пользователем // Компоненты и технологии. 2002. № 9. С. 32-36.
- 13. SOPC Builder User Guide. https://www.altera.com/en\_US/pdfs /literature /ug/ug\_sopc\_builder.pdf
- 14. Ефремов Н.В., Бородин А.А. Инструментальные средства проектирования и отладки систем на программируемых кристаллах компании Altera: учеб. пособие. М.: МГУС, 2012. 150 с.
- 15. Проектирование систем на кристалле на основе ПЛИС. http://e-learning.bmstu.ru/moodle/pluginfile.php/2978/mod\_data/content/880/EVM Lab1.pdf
- 16. Как запрограммировать конфигурационное ПЗУ Altera типа EPCS. http://acvarif.info/artplis/artplis3.html

#### Оглавление

| Лабораторная работа № 1 Изучение протокола передачи данных SPI   | 1   |
|------------------------------------------------------------------|-----|
| Лабораторная работа № 2 Изучение протокола передачи данных I2C   | .15 |
| Лабораторная работа № 3 Изучение протоколов передачи видеоданных |     |
| VGA, DVI и LVDS                                                  | .28 |
| Лабораторная работа № 4 Изучение основ проектирования систем на  |     |
| кристалле на примере процессора Nios II                          | .34 |
| Приложение                                                       | .44 |
| Библиографический список                                         |     |
|                                                                  |     |